home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / CHATZILLA.XPI / bin / chrome / chatzilla.jar / content / chatzilla / handlers.js < prev    next >
Encoding:
JavaScript  |  2005-07-27  |  67.8 KB  |  2,531 lines

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  *
  3.  * ***** BEGIN LICENSE BLOCK *****
  4.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  5.  *
  6.  * The contents of this file are subject to the Mozilla Public License Version
  7.  * 1.1 (the "License"); you may not use this file except in compliance with
  8.  * the License. You may obtain a copy of the License at
  9.  * http://www.mozilla.org/MPL/
  10.  *
  11.  * Software distributed under the License is distributed on an "AS IS" basis,
  12.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13.  * for the specific language governing rights and limitations under the
  14.  * License.
  15.  *
  16.  * The Original Code is ChatZilla.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * New Dimensions Consulting, Inc.
  20.  * Portions created by the Initial Developer are Copyright (C) 1999
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Robert Ginda, rginda@netscape.com, original author
  25.  *   Samuel Sieb, samuel@sieb.net
  26.  *   Chiaki Koufugata chiaki@mozilla.gr.jp UI i18n
  27.  *
  28.  * Alternatively, the contents of this file may be used under the terms of
  29.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  30.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  31.  * in which case the provisions of the GPL or the LGPL are applicable instead
  32.  * of those above. If you wish to allow use of your version of this file only
  33.  * under the terms of either the GPL or the LGPL, and not to allow others to
  34.  * use your version of this file under the terms of the MPL, indicate your
  35.  * decision by deleting the provisions above and replace them with the notice
  36.  * and other provisions required by the GPL or the LGPL. If you do not delete
  37.  * the provisions above, a recipient may use your version of this file under
  38.  * the terms of any one of the MPL, the GPL or the LGPL.
  39.  *
  40.  * ***** END LICENSE BLOCK ***** */
  41.  
  42. window.onresize =
  43. function onresize()
  44. {
  45.     for (var i = 0; i < client.deck.childNodes.length; i++)
  46.         scrollDown(client.deck.childNodes[i], true);
  47. }
  48.  
  49. function onInputFocus()
  50. {
  51. }
  52.  
  53. function showErrorDlg(message)
  54. {
  55.     const XUL_MIME = "application/vnd.mozilla.xul+xml";
  56.     const XUL_KEY = "http://www.mozilla.org/keymaster/" +
  57.                     "gatekeeper/there.is.only.xul";
  58.  
  59.     const TITLE = "ChatZilla run-time error";
  60.     const HEADER = "There was a run-time error with ChatZilla. " +
  61.                    "Please report the following information:";
  62.  
  63.     const OL_JS = "document.getElementById('tb').value = '%S';";
  64.     const TB_STYLE = ' multiline="true" readonly="true"' +
  65.                      ' style="width: 60ex; height: 20em;"';
  66.  
  67.     const ERROR_DLG = '<?xml version="1.0"?>' +
  68.           '<?xml-stylesheet href="chrome://global/skin/" type="text/css"?>' +
  69.           '<dialog xmlns="' + XUL_KEY + '" buttons="accept" ' +
  70.           'title="' + TITLE + '" onload="' + OL_JS + '">' +
  71.           '<label>' + HEADER + '</label><textbox' + TB_STYLE + ' id="tb"/>' +
  72.           '</dialog>';
  73.  
  74.     var content = message.replace(/\n/g, "\\n");
  75.     content = content.replace(/'/g, "\\'").replace(/"/g, """);
  76.     content = content.replace(/</g, "<").replace(/>/g, ">");
  77.     content = ERROR_DLG.replace("%S", content);
  78.     content = encodeURIComponent(content);
  79.     content = "data:" + XUL_MIME + "," + content;
  80.  
  81.     setTimeout(function() {
  82.                    window.openDialog(content, "_blank", "chrome,modal");
  83.                }, 100);
  84. }
  85.  
  86. function onLoad()
  87. {
  88.     dd ("Initializing ChatZilla {");
  89.     try
  90.     {
  91.         init();
  92.     }
  93.     catch (ex)
  94.     {
  95.         dd("caught exception while initializing:\n" + dumpObjectTree(ex));
  96.         showErrorDlg(formatException(ex) + "\n" + dumpObjectTree(ex));
  97.     }
  98.  
  99.     dd("}");
  100.     mainStep();
  101. }
  102.  
  103. function initHandlers()
  104. {
  105.     var node;
  106.     node = document.getElementById("input");
  107.     node.addEventListener("keypress", onInputKeyPress, false);
  108.     node = document.getElementById("multiline-input");
  109.     node.addEventListener("keypress", onMultilineInputKeyPress, false);
  110.     node.active = false;
  111.     node = document.getElementById("security-button");
  112.     node.addEventListener("dblclick", onSecurityIconDblClick, false);
  113.  
  114.     window.onkeypress = onWindowKeyPress;
  115.  
  116.     window.isFocused = false;
  117.     window.addEventListener("focus", onWindowFocus, true);
  118.     window.addEventListener("blur", onWindowBlue, true);
  119.  
  120.     client.inputPopup = null;
  121. }
  122.  
  123. function onClose()
  124. {
  125.     if ("userClose" in client && client.userClose)
  126.         return true;
  127.  
  128.     if (!("getConnectionCount" in client) ||
  129.         client.getConnectionCount() == 0)
  130.     {
  131.         /* if we're not connected to anything, just close the window */
  132.         return true;
  133.     }
  134.  
  135.     /* otherwise, try to close out gracefully */
  136.     var close = true;
  137.     if (client.prefs["warnOnClose"])
  138.     {
  139.         const buttons = ["!yes", "!no"];
  140.         var checkState = { value: true };
  141.         var rv = confirmEx(MSG_CONFIRM_QUIT, buttons, 0, MSG_WARN_ON_EXIT,
  142.                            checkState);
  143.         close = (rv == 0);
  144.         client.prefs["warnOnClose"] = checkState.value;
  145.     }
  146.  
  147.     if (close)
  148.     {
  149.         client.userClose = true;
  150.         display(MSG_CLOSING);
  151.         client.quit(client.userAgent);
  152.     }
  153.     return false;
  154. }
  155.  
  156. function onUnload()
  157. {
  158.     dd("Shutting down ChatZilla.");
  159.     destroy();
  160. }
  161.  
  162. function onNotImplemented()
  163. {
  164.     alert (getMsg("onNotImplementedMsg"));
  165. }
  166.  
  167. /* tab click */
  168. function onTabClick (e, id)
  169. {
  170.     if ((e.which != 1) && (e.which != 2))
  171.         return;
  172.  
  173.     var tbi = document.getElementById (id);
  174.     var view = client.viewsArray[tbi.getAttribute("viewKey")];
  175.  
  176.     if (e.which == 2)
  177.     {
  178.         dispatch("hide", { view: view.source });
  179.         return;
  180.     }
  181.     else if (e.which == 1)
  182.     {
  183.         dispatch("set-current-view", { view: view.source });
  184.     }
  185. }
  186.  
  187. function onMessageViewClick(e)
  188. {
  189.     if ((e.which != 1) && (e.which != 2))
  190.         return true;
  191.  
  192.     var cx = getMessagesContext(null, e.target);
  193.     var command;
  194.  
  195.     if (e.which == 2)
  196.         command = client.prefs["messages.middleClick"];
  197.     else if (e.metaKey || e.altKey)
  198.         command = client.prefs["messages.metaClick"];
  199.     else if (e.ctrlKey)
  200.         command = client.prefs["messages.ctrlClick"];
  201.     else
  202.         command = client.prefs["messages.click"];
  203.  
  204.     if (client.commandManager.isCommandSatisfied(cx, command))
  205.     {
  206.         dispatch(command, cx);
  207.         dispatch("focus-input");
  208.         e.preventDefault();
  209.         return true;
  210.     }
  211.  
  212.     return false;
  213. }
  214.  
  215. function onMouseOver (e)
  216. {
  217.     var i = 0;
  218.     var target = e.target;
  219.     var status = "";
  220.     while (!status && target && i < 5)
  221.     {
  222.         if ("getAttribute" in target)
  223.         {
  224.             status = target.getAttribute("href");
  225.             if (!status)
  226.                 status = target.getAttribute ("statusText");
  227.         }
  228.         ++i;
  229.         target = target.parentNode;
  230.     }
  231.  
  232.     if (status)
  233.     {
  234.         client.status = status;
  235.     }
  236.     else
  237.     {
  238.         if (client && "defaultStatus" in client)
  239.             client.status = client.defaultStatus;
  240.     }
  241. }
  242.  
  243. function onSortCol(sortColName)
  244. {
  245.     var node = document.getElementById(sortColName);
  246.     if (!node)
  247.         return false;
  248.  
  249.     // determine column resource to sort on
  250.     var sortResource = node.getAttribute("resource");
  251.     var sortDirection = node.getAttribute("sortDirection");
  252.  
  253.     if (sortDirection == "ascending")
  254.         sortDirection = "descending";
  255.     else
  256.         sortDirection = "ascending";
  257.  
  258.     sortUserList(node, sortDirection);
  259.  
  260.     return false;
  261. }
  262.  
  263. function onSecurityIconDblClick(e)
  264. {
  265.     if (e.button == 0)
  266.         displayCertificateInfo();
  267. }
  268.  
  269. function onMultilineInputKeyPress (e)
  270. {
  271.     if ((e.ctrlKey || e.metaKey) && e.keyCode == 13)
  272.     {
  273.         /* meta-enter, execute buffer */
  274.         onMultilineSend(e);
  275.     }
  276.     else
  277.     {
  278.         if ((e.ctrlKey || e.metaKey) && e.keyCode == 40)
  279.         {
  280.             /* ctrl/meta-down, switch to single line mode */
  281.             dispatch ("pref multiline false");
  282.         }
  283.     }
  284. }
  285.  
  286. function onMultilineSend(e)
  287. {
  288.     var multiline = document.getElementById("multiline-input");
  289.     e.line = multiline.value;
  290.     if (e.line.search(/\S/) == -1)
  291.         return;
  292.     onInputCompleteLine (e);
  293.     multiline.value = "";
  294. }
  295.  
  296. function onTooltip(event)
  297. {
  298.     const XLinkNS = "http://www.w3.org/1999/xlink";
  299.  
  300.     var tipNode = event.originalTarget;
  301.     var titleText = null;
  302.     var XLinkTitleText = null;
  303.  
  304.     var element = document.tooltipNode;
  305.     while (element)
  306.     {
  307.         if (element.nodeType == Node.ELEMENT_NODE)
  308.         {
  309.             var text;
  310.             if (element.hasAttribute("title"))
  311.                 text = element.getAttribute("title");
  312.             else if (element.hasAttributeNS(XLinkNS, "title"))
  313.                 text = element.getAttributeNS(XLinkNS, "title");
  314.  
  315.             if (text)
  316.             {
  317.                 tipNode.setAttribute("label", text);
  318.                 return true;
  319.             }
  320.         }
  321.  
  322.         element = element.parentNode;
  323.     }
  324.  
  325.     return false;
  326. }
  327.  
  328. function onInputKeyPress (e)
  329. {
  330.     if (client.prefs["outgoing.colorCodes"])
  331.         setTimeout(onInputKeyPressCallback, 100, e.target);
  332.  
  333.     switch (e.keyCode)
  334.     {
  335.         case 9:  /* tab */
  336.             if (!e.ctrlKey && !e.metaKey)
  337.             {
  338.                 onTabCompleteRequest(e);
  339.                 e.preventDefault();
  340.             }
  341.             break;
  342.  
  343.         case 13: /* CR */
  344.             e.line = e.target.value;
  345.             e.target.value = "";
  346.             if (e.line.search(/\S/) == -1)
  347.                 return;
  348.             onInputCompleteLine (e);
  349.             break;
  350.  
  351.         case 38: /* up */
  352.             if (e.ctrlKey || e.metaKey)
  353.             {
  354.                 /* ctrl/meta-up, switch to multi line mode */
  355.                 dispatch ("pref multiline true");
  356.             }
  357.             else
  358.             {
  359.                 if (client.lastHistoryReferenced == -2)
  360.                 {
  361.                     client.lastHistoryReferenced = -1;
  362.                     e.target.value = client.incompleteLine;
  363.                 }
  364.                 else if (client.lastHistoryReferenced <
  365.                          client.inputHistory.length - 1)
  366.                 {
  367.                     e.target.value =
  368.                         client.inputHistory[++client.lastHistoryReferenced];
  369.                 }
  370.             }
  371.             e.preventDefault();
  372.             break;
  373.  
  374.         case 40: /* down */
  375.             if (client.lastHistoryReferenced > 0)
  376.                 e.target.value =
  377.                     client.inputHistory[--client.lastHistoryReferenced];
  378.             else if (client.lastHistoryReferenced == -1)
  379.             {
  380.                 e.target.value = "";
  381.                 client.lastHistoryReferenced = -2;
  382.             }
  383.             else
  384.             {
  385.                 client.lastHistoryReferenced = -1;
  386.                 e.target.value = client.incompleteLine;
  387.             }
  388.             e.preventDefault();
  389.             break;
  390.  
  391.         default:
  392.             client.lastHistoryReferenced = -1;
  393.             client.incompleteLine = e.target.value;
  394.             break;
  395.     }
  396.  
  397. }
  398.  
  399. function onTabCompleteRequest (e)
  400. {
  401.     var elem = document.commandDispatcher.focusedElement;
  402.     var singleInput = document.getElementById("input");
  403.     if (document.getBindingParent(elem) != singleInput)
  404.         return;
  405.  
  406.     var selStart = singleInput.selectionStart;
  407.     var selEnd = singleInput.selectionEnd;
  408.     var line = singleInput.value;
  409.  
  410.     if (!line)
  411.     {
  412.         if ("defaultCompletion" in client.currentObject)
  413.             singleInput.value = client.currentObject.defaultCompletion;
  414.         return;
  415.     }
  416.  
  417.     if (selStart != selEnd)
  418.     {
  419.         /* text is highlighted, just move caret to end and exit */
  420.         singleInput.selectionStart = singleInput.selectionEnd = line.length;
  421.         return;
  422.     }
  423.  
  424.     var wordStart = line.substr(0, selStart).search(/\s\S*$/);
  425.     if (wordStart == -1)
  426.         wordStart = 0;
  427.     else
  428.         ++wordStart;
  429.  
  430.     var wordEnd = line.substr(selStart).search(/\s/);
  431.     if (wordEnd == -1)
  432.         wordEnd = line.length;
  433.     else
  434.         wordEnd += selStart;
  435.  
  436.     if ("performTabMatch" in client.currentObject)
  437.     {
  438.         var word = line.substring(wordStart, wordEnd);
  439.         var wordLower = word.toLowerCase();
  440.         var d = getObjectDetails(client.currentObject);
  441.         if (d.server)
  442.             wordLower = d.server.toLowerCase(word);
  443.  
  444.         var co = client.currentObject;
  445.  
  446.         // We need some special knowledge of how to lower-case strings.
  447.         var lcFn;
  448.         if ("getLCFunction" in co)
  449.             lcFn = co.getLCFunction();
  450.  
  451.         var matches = co.performTabMatch(line, wordStart, wordEnd, wordLower,
  452.                                          selStart, lcFn);
  453.         /* if we get null back, we're supposed to fail silently */
  454.         if (!matches)
  455.             return;
  456.  
  457.         var doubleTab = false;
  458.         var date = new Date();
  459.         if ((date - client.lastTabUp) <= client.DOUBLETAB_TIME)
  460.             doubleTab = true;
  461.         else
  462.             client.lastTabUp = date;
  463.  
  464.         if (doubleTab)
  465.         {
  466.             /* if the user hit tab twice quickly, */
  467.             if (matches.length > 0)
  468.             {
  469.                 /* then list possible completions, */
  470.                 display(getMsg(MSG_FMT_MATCHLIST,
  471.                                [matches.length, word,
  472.                                 matches.join(MSG_COMMASP)]));
  473.             }
  474.             else
  475.             {
  476.                 /* or display an error if there are none. */
  477.                 display(getMsg(MSG_ERR_NO_MATCH, word), MT_ERROR);
  478.             }
  479.         }
  480.         else if (matches.length >= 1)
  481.         {
  482.             var match;
  483.             if (matches.length == 1)
  484.                 match = matches[0];
  485.             else
  486.                 match = getCommonPfx(matches, lcFn);
  487.             singleInput.value = line.substr(0, wordStart) + match +
  488.                     line.substr(wordEnd);
  489.             if (wordEnd < line.length)
  490.             {
  491.                 /* if the word we completed was in the middle if the line
  492.                  * then move the cursor to the end of the completed word. */
  493.                 var newpos = wordStart + match.length;
  494.                 if (matches.length == 1)
  495.                 {
  496.                     /* word was fully completed, move one additional space */
  497.                     ++newpos;
  498.                 }
  499.                 singleInput.selectionEnd = e.target.selectionStart = newpos;
  500.             }
  501.         }
  502.     }
  503.  
  504. }
  505.  
  506. function onWindowKeyPress (e)
  507. {
  508.     var code = Number(e.keyCode);
  509.     var w;
  510.     var newOfs;
  511.     var userList = document.getElementById("user-list");
  512.     var elemFocused = document.commandDispatcher.focusedElement;
  513.  
  514.     switch (code)
  515.     {
  516.         case 9: /* tab */
  517.             if (e.ctrlKey || e.metaKey)
  518.             {
  519.                 cycleView(e.shiftKey ? -1: 1);
  520.                 e.preventDefault();
  521.             }
  522.             break;
  523.  
  524.         case 112: /* F1 */
  525.         case 113: /* ... */
  526.         case 114:
  527.         case 115:
  528.         case 116:
  529.         case 117:
  530.         case 118:
  531.         case 119:
  532.         case 120:
  533.         case 121: /* F10 */
  534.             var modifier = (e.ctrlKey || e.shiftKey || e.Altkey || e.metaKey);
  535.             var idx = code - 112;
  536.             if (!modifier && (idx in client.viewsArray) &&
  537.                 client.viewsArray[idx].source)
  538.             {
  539.                 var newView = client.viewsArray[idx].source;
  540.                 dispatch("set-current-view", { view: newView });
  541.                 e.preventDefault();
  542.             }
  543.             break;
  544.  
  545.         case 33: /* pgup */
  546.             if (e.ctrlKey)
  547.             {
  548.                 cycleView(-1);
  549.                 e.preventDefault();
  550.                 break;
  551.             }
  552.  
  553.             if (elemFocused == userList)
  554.                 break;
  555.  
  556.             w = client.currentFrame;
  557.             newOfs = w.pageYOffset - (w.innerHeight * 0.75);
  558.             if (newOfs > 0)
  559.                 w.scrollTo (w.pageXOffset, newOfs);
  560.             else
  561.                 w.scrollTo (w.pageXOffset, 0);
  562.             e.preventDefault();
  563.             break;
  564.  
  565.         case 34: /* pgdn */
  566.             if (e.ctrlKey)
  567.             {
  568.                 cycleView(1);
  569.                 e.preventDefault();
  570.                 break;
  571.             }
  572.  
  573.             if (elemFocused == userList)
  574.                 break;
  575.  
  576.             w = client.currentFrame;
  577.             newOfs = w.pageYOffset + (w.innerHeight * 0.75);
  578.             if (newOfs < (w.innerHeight + w.pageYOffset))
  579.                 w.scrollTo (w.pageXOffset, newOfs);
  580.             else
  581.                 w.scrollTo (w.pageXOffset, (w.innerHeight + w.pageYOffset));
  582.             e.preventDefault();
  583.             break;
  584.     }
  585. }
  586.  
  587. function onWindowFocus(e)
  588. {
  589.     window.isFocused = true;
  590. }
  591.  
  592. function onWindowBlue(e)
  593. {
  594.     window.isFocused = false;
  595. }
  596.  
  597. function onInputCompleteLine(e)
  598. {
  599.     if (!client.inputHistory.length || client.inputHistory[0] != e.line)
  600.         client.inputHistory.unshift (e.line);
  601.  
  602.     if (client.inputHistory.length > client.MAX_HISTORY)
  603.         client.inputHistory.pop();
  604.  
  605.     client.lastHistoryReferenced = -1;
  606.     client.incompleteLine = "";
  607.  
  608.     if (e.line[0] == client.COMMAND_CHAR)
  609.     {
  610.         if (client.prefs["outgoing.colorCodes"])
  611.             e.line = replaceColorCodes(e.line);
  612.         dispatch(e.line.substr(1), null, true);
  613.     }
  614.     else /* plain text */
  615.     {
  616.         /* color codes */
  617.         if (client.prefs["outgoing.colorCodes"])
  618.             e.line = replaceColorCodes(e.line);
  619.         client.sayToCurrentTarget (e.line);
  620.     }
  621. }
  622.  
  623. function onNotifyTimeout ()
  624. {
  625.     for (var n in client.networks)
  626.     {
  627.         var net = client.networks[n];
  628.         if (net.isConnected()) {
  629.             if (net.prefs["notifyList"].length > 0) {
  630.                 var isonList = client.networks[n].prefs["notifyList"];
  631.                 net.primServ.sendData ("ISON " + isonList.join(" ") + "\n");
  632.             } else {
  633.                 /* if the notify list is empty, just send a ping to see if we're
  634.                  * alive. */
  635.                 net.primServ.sendData ("PING :ALIVECHECK\n");
  636.             }
  637.         }
  638.     }
  639. }
  640.  
  641. var lastWhoCheckTime = new Date();
  642. var lastWhoCheckChannel = null;
  643.  
  644. function onWhoTimeout()
  645. {
  646.     lastWhoCheckTime = new Date();
  647.     var checkNext = (lastWhoCheckChannel == null);
  648.  
  649.     for (var n in client.networks)
  650.     {
  651.         var net = client.networks[n];
  652.         if (net.isConnected())
  653.         {
  654.             for (var c in net.primServ.channels)
  655.             {
  656.                 var chan = net.primServ.channels[c];
  657.  
  658.                 if (checkNext && chan.active &&
  659.                     chan.getUsersLength() < client.prefs["autoAwayCap"])
  660.                 {
  661.                     net.primServ.LIGHTWEIGHT_WHO = true;
  662.                     net.primServ.sendData("WHO " + chan.encodedName + "\n");
  663.                     lastWhoCheckChannel = chan;
  664.                     return;
  665.                 }
  666.  
  667.                 if (chan == lastWhoCheckChannel)
  668.                     checkNext = true;
  669.             }
  670.         }
  671.     }
  672.  
  673.     if (lastWhoCheckChannel)
  674.     {
  675.         lastWhoCheckChannel = null;
  676.         onWhoTimeout();
  677.     }
  678. }
  679.  
  680. function onInputKeyPressCallback (el)
  681. {
  682.     function doPopup(popup)
  683.     {
  684.         if (client.inputPopup && client.inputPopup != popup)
  685.             client.inputPopup.hidePopup();
  686.  
  687.         client.inputPopup = popup;
  688.         if (popup)
  689.         {
  690.             if (el.nodeName == "textbox")
  691.             {
  692.                 popup.showPopup(el, -1, -1, "tooltip", "topleft", "bottomleft");
  693.             }
  694.             else
  695.             {
  696.                 var box = el.ownerDocument.getBoxObjectFor(el);
  697.                 var pos = { x: client.mainWindow.screenX + box.screenX + 5,
  698.                             y: client.mainWindow.screenY + box.screenY + box.height + 25 };
  699.                 popup.moveTo(pos.x, pos.y);
  700.                 popup.showPopup(el, 0, 0, "tooltip");
  701.             }
  702.         }
  703.     }
  704.  
  705.     var text = " " + el.value.substr(0, el.selectionStart);
  706.     if (el.selectionStart != el.selectionEnd)
  707.         text = "";
  708.  
  709.     if (text.match(/[^%]%C[0-9]{0,2},?[0-9]{0,2}$/))
  710.         doPopup(document.getElementById("colorTooltip"));
  711.     else if (text.match(/[^%]%$/))
  712.         doPopup(document.getElementById("percentTooltip"));
  713.     else
  714.         doPopup(null);
  715. }
  716.  
  717. /* 'private' function, should only be used from inside */
  718. CIRCChannel.prototype._addUserToGraph =
  719. function my_addtograph (user)
  720. {
  721.     if (!user.TYPE)
  722.         dd (getStackTrace());
  723.  
  724.     client.rdf.Assert (this.getGraphResource(), client.rdf.resChanUser,
  725.                        user.getGraphResource(), true);
  726.  
  727. }
  728.  
  729. /* 'private' function, should only be used from inside */
  730. CIRCChannel.prototype._removeUserFromGraph =
  731. function my_remfgraph (user)
  732. {
  733.  
  734.     client.rdf.Unassert (this.getGraphResource(), client.rdf.resChanUser,
  735.                          user.getGraphResource());
  736.  
  737. }
  738.  
  739. CIRCServer.prototype.CTCPHelpClientinfo =
  740. function serv_ccinfohelp()
  741. {
  742.     return MSG_CTCPHELP_CLIENTINFO;
  743. }
  744.  
  745. CIRCServer.prototype.CTCPHelpAction =
  746. function serv_ccinfohelp()
  747. {
  748.     return MSG_CTCPHELP_ACTION;
  749. }
  750.  
  751. CIRCServer.prototype.CTCPHelpTime =
  752. function serv_ccinfohelp()
  753. {
  754.     return MSG_CTCPHELP_TIME;
  755. }
  756.  
  757. CIRCServer.prototype.CTCPHelpVersion =
  758. function serv_ccinfohelp()
  759. {
  760.     return MSG_CTCPHELP_VERSION;
  761. }
  762.  
  763. CIRCServer.prototype.CTCPHelpSource =
  764. function serv_csrchelp()
  765. {
  766.     return MSG_CTCPHELP_SOURCE;
  767. }
  768.  
  769. CIRCServer.prototype.CTCPHelpOs =
  770. function serv_oshelp()
  771. {
  772.     return MSG_CTCPHELP_OS;
  773. }
  774.  
  775. CIRCServer.prototype.CTCPHelpHost =
  776. function serv_hosthelp()
  777. {
  778.     return MSG_CTCPHELP_HOST;
  779. }
  780.  
  781. CIRCServer.prototype.CTCPHelpPing =
  782. function serv_ccinfohelp()
  783. {
  784.     return MSG_CTCPHELP_PING;
  785. }
  786.  
  787. CIRCServer.prototype.CTCPHelpDcc =
  788. function serv_ccinfohelp()
  789. {
  790.     return MSG_CTCPHELP_DCC;
  791. }
  792.  
  793. CIRCNetwork.prototype.onInit =
  794. function net_oninit ()
  795. {
  796.     this.logFile = null;
  797.     this.lastServer = null;
  798. }
  799.  
  800. CIRCNetwork.prototype.onInfo =
  801. function my_netinfo (e)
  802. {
  803.     this.display (e.msg, "INFO");
  804. }
  805.  
  806. CIRCNetwork.prototype.onUnknown =
  807. function my_unknown (e)
  808. {
  809.     if ("pendingWhoisLines" in e.server)
  810.     {
  811.         /* whois lines always have the nick in param 2 */
  812.         e.user = new CIRCUser(e.server, null, e.params[2]);
  813.  
  814.         e.destMethod = "onUnknownWhois";
  815.         e.destObject = this;
  816.         return;
  817.     }
  818.  
  819.     e.params.shift(); /* remove the code */
  820.     e.params.shift(); /* and the dest. nick (always me) */
  821.  
  822.     // Handle random IRC numerics automatically.
  823.     var msg = getMsg("msg.err.irc." + e.code, null, "");
  824.     if (msg)
  825.     {
  826.         if (arrayIndexOf(e.server.channelTypes, e.params[0][0]) != -1)
  827.         {
  828.             // Message about a channel (e.g. join failed).
  829.             e.channel = new CIRCChannel(e.server, null, e.params[0]);
  830.         }
  831.  
  832.         // Check for /knock support for the +i message.
  833.         if (((e.code == 471) || (e.code == 473) || (e.code == 475)) &&
  834.             ("knock" in e.server.servCmds))
  835.         {
  836.             var args = [msg, e.channel.unicodeName,
  837.                         "knock " + e.channel.unicodeName];
  838.             msg = getMsg("msg.err.irc." + e.code + ".knock", args, "");
  839.             client.munger.entries[".inline-buttons"].enabled = true;
  840.             this.display(msg);
  841.             client.munger.entries[".inline-buttons"].enabled = false;
  842.         }
  843.         else
  844.         {
  845.             this.display(msg);
  846.         }
  847.  
  848.         if (e.channel)
  849.         {
  850.             if (e.channel.busy)
  851.             {
  852.                 e.channel.busy = false;
  853.                 updateProgress();
  854.             }
  855.         }
  856.         else
  857.         {
  858.             // Network type error?
  859.             if (this.busy)
  860.             {
  861.                 this.busy = false;
  862.                 updateProgress();
  863.             }
  864.         }
  865.         return;
  866.     }
  867.  
  868.         /* if it looks like some kind of "end of foo" code, and we don't
  869.          * already have a mapping for it, make one up */
  870.     var length = e.params.length;
  871.     if (!(e.code in client.responseCodeMap) &&
  872.         (e.params[length - 1].search (/^end of/i) != -1))
  873.     {
  874.         client.responseCodeMap[e.code] = "---";
  875.     }
  876.  
  877.     this.display(toUnicode(e.params.join(" "), this), e.code.toUpperCase());
  878. }
  879.  
  880. CIRCNetwork.prototype.on001 = /* Welcome! */
  881. CIRCNetwork.prototype.on002 = /* your host is */
  882. CIRCNetwork.prototype.on003 = /* server born-on date */
  883. CIRCNetwork.prototype.on004 = /* server id */
  884. CIRCNetwork.prototype.on005 = /* server features */
  885. CIRCNetwork.prototype.on250 = /* highest connection count */
  886. CIRCNetwork.prototype.on251 = /* users */
  887. CIRCNetwork.prototype.on252 = /* opers online (in params[2]) */
  888. CIRCNetwork.prototype.on254 = /* channels found (in params[2]) */
  889. CIRCNetwork.prototype.on255 = /* link info */
  890. CIRCNetwork.prototype.on265 = /* local user details */
  891. CIRCNetwork.prototype.on266 = /* global user details */
  892. CIRCNetwork.prototype.on375 = /* start of MOTD */
  893. CIRCNetwork.prototype.on372 = /* MOTD line */
  894. CIRCNetwork.prototype.on376 = /* end of MOTD */
  895. CIRCNetwork.prototype.on422 = /* no MOTD */
  896. function my_showtonet (e)
  897. {
  898.     var p = (3 in e.params) ? e.params[2] + " " : "";
  899.     var str = "";
  900.  
  901.     switch (e.code)
  902.     {
  903.         case "004":
  904.         case "005":
  905.             str = e.params.slice(3).join(" ");
  906.             break;
  907.  
  908.         case "001":
  909.             // Code moved to lower down to speed this bit up. :)
  910.  
  911.             var cmdary = this.prefs["autoperform"];
  912.             for (var i = 0; i < cmdary.length; ++i)
  913.                 this.dispatch(cmdary[i])
  914.  
  915.             if (this.prefs["away"])
  916.                 this.dispatch("away", { reason: this.prefs["away"] });
  917.  
  918.             if (this.lastServer)
  919.             {
  920.                 var c, u;
  921.  
  922.                 // Re-home channels and users only if nessessary.
  923.                 if (this.lastServer != this.primServ)
  924.                 {
  925.                     for (c in this.lastServer.channels)
  926.                         this.lastServer.channels[c].rehome(this.primServ);
  927.                     for (u in this.lastServer.users)
  928.                         this.lastServer.users[u].rehome(this.primServ);
  929.  
  930.                     // This makes sure we have the *right* me object.
  931.                     this.primServ.me.rehome(this.primServ);
  932.                 }
  933.  
  934.                 // Re-join channels from previous connection.
  935.                 // (note they're all on .primServ now, not .lastServer)
  936.                 for (c in this.primServ.channels)
  937.                 {
  938.                     var chan = this.primServ.channels[c];
  939.                     if (chan.joined)
  940.                         chan.join(chan.mode.key);
  941.                 }
  942.             }
  943.             this.lastServer = this.primServ;
  944.  
  945.             if ("pendingURLs" in this)
  946.             {
  947.                 var url = this.pendingURLs.pop();
  948.                 while (url)
  949.                 {
  950.                     gotoIRCURL(url);
  951.                     url = this.pendingURLs.pop();
  952.                 }
  953.                 delete this.pendingURLs;
  954.             }
  955.  
  956.             // Update everything.
  957.             // Welcome to history.
  958.             if (client.globalHistory)
  959.                 client.globalHistory.addPage(this.getURL());
  960.             updateTitle(this);
  961.             this.updateHeader();
  962.             client.updateHeader();
  963.             updateSecurityIcon();
  964.             updateStalkExpression(this);
  965.  
  966.             // Do this after the JOINs, so they are quicker.
  967.             // This is not time-critical code.
  968.             if (jsenv.HAS_SERVER_SOCKETS && client.prefs["dcc.enabled"] &&
  969.                 this.prefs["dcc.useServerIP"])
  970.             {
  971.                 var delayFn = function(t) {
  972.                     // This is the quickest way to get out host/IP.
  973.                     t.pendingUserhostReply = true;
  974.                     t.primServ.sendData("USERHOST " +
  975.                                         t.primServ.me.encodedName + "\n");
  976.                 };
  977.                 setTimeout(delayFn, 1000 * Math.random(), this);
  978.             }
  979.  
  980.             // Had some collision during connect.
  981.             if (this.primServ.me.unicodeName != this.prefs["nickname"])
  982.             {
  983.                 this.reclaimLeft = this.RECLAIM_TIMEOUT;
  984.                 this.reclaimName();
  985.             }
  986.  
  987.             str = e.decodeParam(2);
  988.             if ("onLogin" in this)
  989.             {
  990.                 ev = new CEvent("network", "login", this, "onLogin");
  991.                 client.eventPump.addEvent(ev);
  992.             }
  993.             break;
  994.  
  995.         case "376": /* end of MOTD */
  996.         case "422": /* no MOTD */
  997.             this.busy = false;
  998.             updateProgress();
  999.             /* no break */
  1000.  
  1001.         case "372":
  1002.         case "375":
  1003.         case "376":
  1004.             if (this.IGNORE_MOTD)
  1005.                 return;
  1006.             /* no break */
  1007.  
  1008.         default:
  1009.             str = e.decodeParam(e.params.length - 1);
  1010.             break;
  1011.     }
  1012.  
  1013.     this.displayHere(p + str, e.code.toUpperCase());
  1014.  
  1015. }
  1016.  
  1017. CIRCNetwork.prototype.onUnknownCTCPReply =
  1018. function my_ctcprunk (e)
  1019. {
  1020.     this.display(getMsg(MSG_FMT_CTCPREPLY,
  1021.                         [toUnicode(e.CTCPCode, this),
  1022.                          toUnicode(e.CTCPData, this), e.user.unicodeName]),
  1023.                  "CTCP_REPLY", e.user, e.server.me, this);
  1024. }
  1025.  
  1026. CIRCNetwork.prototype.onNotice =
  1027. function my_notice (e)
  1028. {
  1029.     this.display(e.decodeParam(2), "NOTICE", this, e.server.me);
  1030. }
  1031.  
  1032. /* userhost reply */
  1033. CIRCNetwork.prototype.on302 =
  1034. function my_302(e)
  1035. {
  1036.     if (jsenv.HAS_SERVER_SOCKETS && client.prefs["dcc.enabled"] &&
  1037.         this.prefs["dcc.useServerIP"] && ("pendingUserhostReply" in this))
  1038.     {
  1039.         var me = new RegExp("^" + this.primServ.me.encodedName + "\\*?=", "i");
  1040.         if (e.params[2].match(me))
  1041.             client.dcc.addHost(this.primServ.me.host, true);
  1042.  
  1043.         delete this.pendingUserhostReply;
  1044.         return true;
  1045.     }
  1046.  
  1047.     e.destMethod = "onUnknown";
  1048.     e.destObject = this;
  1049.  
  1050.     return true;
  1051. }
  1052.  
  1053. CIRCNetwork.prototype.on303 = /* ISON (aka notify) reply */
  1054. function my_303 (e)
  1055. {
  1056.     function lower(text)
  1057.     {
  1058.         return e.server.toLowerCase(text);
  1059.     };
  1060.  
  1061.     var onList = new Array();
  1062.     // split() gives an array of one item ("") when splitting "", which we
  1063.     // don't want, so only do the split if there's something to split.
  1064.     if (e.params[2])
  1065.         onList = stringTrim(e.server.toLowerCase(e.params[2])).split(/\s+/);
  1066.     var offList = new Array();
  1067.     var newArrivals = new Array();
  1068.     var newDepartures = new Array();
  1069.     var o = getObjectDetails(client.currentObject);
  1070.     var displayTab;
  1071.     var i;
  1072.  
  1073.     if ("network" in o && o.network == this && client.currentObject != this)
  1074.         displayTab = client.currentObject;
  1075.  
  1076.     for (i = 0; i < this.prefs["notifyList"].length; i++)
  1077.     {
  1078.         if (!arrayContains(onList, lower(this.prefs["notifyList"][i])))
  1079.             /* user is not on */
  1080.             offList.push(lower(this.prefs["notifyList"][i]));
  1081.     }
  1082.  
  1083.     if ("onList" in this)
  1084.     {
  1085.         for (i in onList)
  1086.             if (!arrayContains(this.onList, onList[i]))
  1087.                 /* we didn't know this person was on */
  1088.                 newArrivals.push(onList[i]);
  1089.     }
  1090.     else
  1091.         this.onList = newArrivals = onList;
  1092.  
  1093.     if ("offList" in this)
  1094.     {
  1095.         for (i in offList)
  1096.             if (!arrayContains(this.offList, offList[i]))
  1097.                 /* we didn't know this person was off */
  1098.                 newDepartures.push(offList[i]);
  1099.     }
  1100.     else
  1101.         this.offList = newDepartures = offList;
  1102.  
  1103.     if (newArrivals.length > 0)
  1104.     {
  1105.         this.displayHere (arraySpeak (newArrivals, "is", "are") +
  1106.                           " online.", "NOTIFY-ON");
  1107.         if (displayTab)
  1108.             displayTab.displayHere (arraySpeak (newArrivals, "is", "are") +
  1109.                                     " online.", "NOTIFY-ON");
  1110.     }
  1111.  
  1112.     if (newDepartures.length > 0)
  1113.     {
  1114.         this.displayHere (arraySpeak (newDepartures, "is", "are") +
  1115.                           " offline.", "NOTIFY-OFF");
  1116.         if (displayTab)
  1117.             displayTab.displayHere (arraySpeak (newDepartures, "is", "are") +
  1118.                                     " offline.", "NOTIFY-OFF");
  1119.     }
  1120.  
  1121.     this.onList = onList;
  1122.     this.offList = offList;
  1123.  
  1124. }
  1125.  
  1126. /* away off reply */
  1127. CIRCNetwork.prototype.on305 =
  1128. function my_305(e)
  1129. {
  1130.     this.display(MSG_AWAY_OFF);
  1131.  
  1132.     return true;
  1133. }
  1134.  
  1135. /* away on reply */
  1136. CIRCNetwork.prototype.on306 =
  1137. function my_306(e)
  1138. {
  1139.     this.display(getMsg(MSG_AWAY_ON, this.prefs["away"]));
  1140.  
  1141.     return true;
  1142. }
  1143.  
  1144.  
  1145. CIRCNetwork.prototype.on263 = /* 'try again' */
  1146. function my_263 (e)
  1147. {
  1148.     /* Urgh, this one's a pain. We need to abort whatever we tried, and start
  1149.      * it again if appropriate.
  1150.      *
  1151.      * Known causes of this message:
  1152.      *   - LIST, with or without a parameter.
  1153.      */
  1154.  
  1155.     if (("_list" in this) && !this._list.done && (this._list.count == 0))
  1156.     {
  1157.         // We attempted a LIST, and we think it failed. :)
  1158.         this._list.done = true;
  1159.         this._list.error = e.decodeParam(2);
  1160.         // Return early for this one if we're saving it.
  1161.         if ("saveTo" in this._list)
  1162.             return true;
  1163.     }
  1164.  
  1165.     e.destMethod = "onUnknown";
  1166.     e.destObject = this;
  1167.     return true;
  1168. }
  1169.  
  1170. CIRCNetwork.prototype.list =
  1171. function my_list(word, file)
  1172. {
  1173.     if (("_list" in this) && !this._list.done)
  1174.         return false;
  1175.  
  1176.     this._list = new Array();
  1177.     this._list.string = word;
  1178.     this._list.regexp = null;
  1179.     this._list.done = false;
  1180.     this._list.count = 0;
  1181.     if (file)
  1182.         this._list.saveTo = file;
  1183.  
  1184.     if (word instanceof RegExp)
  1185.     {
  1186.         this._list.regexp = word;
  1187.         this._list.string = "";
  1188.         word = "";
  1189.     }
  1190.  
  1191.     if (word)
  1192.         this.primServ.sendData("LIST " + fromUnicode(word, this) + "\n");
  1193.     else
  1194.         this.primServ.sendData("LIST\n");
  1195.  
  1196.     return true;
  1197. }
  1198.  
  1199. CIRCNetwork.prototype.listInit =
  1200. function my_list_init ()
  1201. {
  1202.     const NORMAL_FILE_TYPE = Components.interfaces.nsIFile.NORMAL_FILE_TYPE;
  1203.  
  1204.     function checkEndList (network)
  1205.     {
  1206.         if (network._list.count == network._list.lastLength)
  1207.         {
  1208.             network.on323();
  1209.         }
  1210.         else
  1211.         {
  1212.             network._list.lastLength = network._list.count;
  1213.             network._list.endTimeout =
  1214.                 setTimeout(checkEndList, 5000, network);
  1215.         }
  1216.     }
  1217.  
  1218.     function outputList (network)
  1219.     {
  1220.         const CHUNK_SIZE = 5;
  1221.         var list = network._list;
  1222.         if (list.length > list.displayed)
  1223.         {
  1224.             var start = list.displayed;
  1225.             var end = list.length;
  1226.             if (end - start > CHUNK_SIZE)
  1227.                 end = start + CHUNK_SIZE;
  1228.             for (var i = start; i < end; ++i)
  1229.                 network.displayHere (getMsg(MSG_FMT_CHANLIST, list[i]), "322");
  1230.             list.displayed = end;
  1231.         }
  1232.         if (list.done && (list.displayed == list.length))
  1233.         {
  1234.             if (list.event323)
  1235.             {
  1236.                 var length = list.event323.params.length;
  1237.                 network.displayHere (list.event323.params[length - 1], "323");
  1238.             }
  1239.             network.displayHere(getMsg(MSG_LIST_END,
  1240.                                        [list.displayed, list.count]));
  1241.             delete network._list;
  1242.         }
  1243.         else
  1244.         {
  1245.             setTimeout(outputList, 250, network);
  1246.         }
  1247.     }
  1248.  
  1249.     if (!("_list" in this))
  1250.     {
  1251.         this._list = new Array();
  1252.         this._list.string = MSG_UNKNOWN;
  1253.         this._list.regexp = null;
  1254.         this._list.done = false;
  1255.         this._list.count = 0;
  1256.     }
  1257.  
  1258.     if ("saveTo" in this._list)
  1259.     {
  1260.         var file = new LocalFile(this._list.saveTo);
  1261.         if (!file.localFile.exists())
  1262.         {
  1263.             // futils.umask may be 0022. Result is 0644.
  1264.             file.localFile.create(NORMAL_FILE_TYPE, 0666 & ~futils.umask);
  1265.         }
  1266.         this._list.file = new LocalFile(file.localFile, ">");
  1267.     }
  1268.     else
  1269.     {
  1270.         this._list.displayed = 0;
  1271.         if (client.currentObject != this)
  1272.             display (getMsg(MSG_LIST_REROUTED, this.unicodeName));
  1273.         setTimeout(outputList, 250, this);
  1274.     }
  1275.     this._list.lastLength = 0;
  1276.     this._list.endTimeout = setTimeout(checkEndList, 5000, this);
  1277. }
  1278.  
  1279. CIRCNetwork.prototype.on321 = /* LIST reply header */
  1280. function my_321 (e)
  1281. {
  1282.  
  1283.     this.listInit();
  1284.  
  1285.     if (!("saveTo" in this._list))
  1286.         this.displayHere (e.params[2] + " " + e.params[3], "321");
  1287. }
  1288.  
  1289. CIRCNetwork.prototype.on323 = /* end of LIST reply */
  1290. function my_323 (e)
  1291. {
  1292.     if (this._list.endTimeout)
  1293.     {
  1294.         clearTimeout(this._list.endTimeout);
  1295.         delete this._list.endTimeout;
  1296.     }
  1297.     if (("file" in this._list))
  1298.         this._list.file.close();
  1299.  
  1300.     this._list.done = true;
  1301.     this._list.event323 = e;
  1302. }
  1303.  
  1304. CIRCNetwork.prototype.on322 = /* LIST reply */
  1305. function my_listrply (e)
  1306. {
  1307.     if (!("_list" in this) || !("lastLength" in this._list))
  1308.         this.listInit();
  1309.  
  1310.     ++this._list.count;
  1311.  
  1312.     var chanName = e.decodeParam(2);
  1313.     var topic = e.decodeParam(4);
  1314.     if (!this._list.regexp || chanName.match(this._list.regexp) ||
  1315.         topic.match(this._list.regexp))
  1316.     {
  1317.         if (!("file" in this._list))
  1318.         {
  1319.             this._list.push([chanName, e.params[3], topic]);
  1320.         }
  1321.         else
  1322.         {
  1323.             this._list.file.write(fromUnicode(chanName, "UTF-8") + " " +
  1324.                                   e.params[3] + " " +
  1325.                                   fromUnicode(topic, "UTF-8") + "\n");
  1326.         }
  1327.     }
  1328. }
  1329.  
  1330. CIRCNetwork.prototype.on401 =
  1331. function my_401 (e)
  1332. {
  1333.     var target = e.server.toLowerCase(e.params[2]);
  1334.     if (target in this.users && "messages" in this.users[target])
  1335.     {
  1336.         this.users[target].displayHere(e.params[3]);
  1337.     }
  1338.     else if (target in this.primServ.channels &&
  1339.              "messages" in this.primServ.channels[target])
  1340.     {
  1341.         this.primServ.channels[target].displayHere(e.params[3]);
  1342.     }
  1343.     else
  1344.     {
  1345.         display(toUnicode(e.params[3], this));
  1346.     }
  1347. }
  1348.  
  1349. /* 464; "invalid or missing password", occurs as a reply to both OPER and
  1350.  * sometimes initially during user registration. */
  1351. CIRCNetwork.prototype.on464 =
  1352. function my_464(e)
  1353. {
  1354.     if (this.state == NET_CONNECTING)
  1355.     {
  1356.         // If we are in the process of connecting we are needing a login
  1357.         // password, subtly different from after user registration.
  1358.         this.display(MSG_ERR_IRC_464_LOGIN);
  1359.     }
  1360.     else
  1361.     {
  1362.         e.destMethod = "onUnknown";
  1363.         e.destObject = this;
  1364.     }
  1365. }
  1366.  
  1367. /* end of WHO */
  1368. CIRCNetwork.prototype.on315 =
  1369. function my_315 (e)
  1370. {
  1371.     var matches;
  1372.     if ("whoMatches" in this)
  1373.         matches = this.whoMatches;
  1374.     else
  1375.         matches = 0;
  1376.  
  1377.     if ("pendingWhoReply" in this)
  1378.         this.display(getMsg(MSG_WHO_END, [e.params[2], matches]), e.code);
  1379.  
  1380.     if ("whoUpdates" in this)
  1381.     {
  1382.         for (var c in this.whoUpdates)
  1383.         {
  1384.             for (var i = 0; i < this.whoUpdates[c].length; i++)
  1385.                 this.whoUpdates[c][i].updateGraphResource();
  1386.             this.primServ.channels[c].updateUsers(this.whoUpdates[c]);
  1387.         }
  1388.         delete this.whoUpdates;
  1389.     }
  1390.  
  1391.     delete this.pendingWhoReply;
  1392.     delete this.whoMatches;
  1393. }
  1394.  
  1395. CIRCNetwork.prototype.on352 =
  1396. function my_352 (e)
  1397. {
  1398.     //0-352 1-rginda_ 2-#chatzilla 3-chatzilla 4-h-64-236-139-254.aoltw.net
  1399.     //5-irc.mozilla.org 6-rginda 7-H
  1400.     if ("pendingWhoReply" in this)
  1401.     {
  1402.         var status;
  1403.         if (e.user.isAway)
  1404.             status = MSG_GONE;
  1405.         else
  1406.             status = MSG_HERE;
  1407.  
  1408.         this.display(getMsg(MSG_WHO_MATCH,
  1409.                             [e.params[6], e.params[3], e.params[4],
  1410.                              e.user.desc, status, e.decodeParam(2),
  1411.                              e.params[5], e.user.hops]), e.code, e.user);
  1412.     }
  1413.  
  1414.     updateTitle(e.user);
  1415.     if ("whoMatches" in this)
  1416.         ++this.whoMatches;
  1417.     else
  1418.         this.whoMatches = 1;
  1419.  
  1420.     if (!("whoUpdates" in this))
  1421.         this.whoUpdates = new Object();
  1422.  
  1423.     if (e.userHasChanges)
  1424.     {
  1425.         for (var c in e.server.channels)
  1426.         {
  1427.             var chan = e.server.channels[c];
  1428.             if (chan.active && (e.user.canonicalName in chan.users))
  1429.             {
  1430.                 if (!(c in this.whoUpdates))
  1431.                     this.whoUpdates[c] = new Array();
  1432.                 this.whoUpdates[c].push(chan.users[e.user.canonicalName]);
  1433.             }
  1434.         }
  1435.     }
  1436. }
  1437.  
  1438. CIRCNetwork.prototype.on311 = /* whois name */
  1439. CIRCNetwork.prototype.on319 = /* whois channels */
  1440. CIRCNetwork.prototype.on312 = /* whois server */
  1441. CIRCNetwork.prototype.on317 = /* whois idle time */
  1442. CIRCNetwork.prototype.on318 = /* whois end of whois*/
  1443. CIRCNetwork.prototype.onUnknownWhois = /* misc whois line */
  1444. function my_whoisreply (e)
  1445. {
  1446.     var text = "egads!";
  1447.     var nick = e.params[2];
  1448.     var user;
  1449.  
  1450.     if (e.user)
  1451.     {
  1452.         user = e.user;
  1453.         nick = user.unicodeName;
  1454.     }
  1455.  
  1456.     switch (Number(e.code))
  1457.     {
  1458.         case 311:
  1459.             text = getMsg(MSG_WHOIS_NAME,
  1460.                           [nick, e.params[3], e.params[4],
  1461.                            e.decodeParam(6)]);
  1462.             break;
  1463.  
  1464.         case 319:
  1465.             var ary = stringTrim(e.decodeParam(3)).split(" ");
  1466.             text = getMsg(MSG_WHOIS_CHANNELS, [nick, arraySpeak(ary)]);
  1467.             break;
  1468.  
  1469.         case 312:
  1470.             text = getMsg(MSG_WHOIS_SERVER,
  1471.                           [nick, e.params[3], e.params[4]]);
  1472.             break;
  1473.  
  1474.         case 317:
  1475.             text = getMsg(MSG_WHOIS_IDLE,
  1476.                           [nick, formatDateOffset(Number(e.params[3])),
  1477.                           new Date(Number(e.params[4]) * 1000)]);
  1478.             break;
  1479.  
  1480.         case 318:
  1481.             text = getMsg(MSG_WHOIS_END, nick);
  1482.             if (user)
  1483.                 user.updateHeader();
  1484.             break;
  1485.  
  1486.         default:
  1487.             text = toUnicode(e.params.splice(2, e.params.length).join(" "),
  1488.                              this);
  1489.     }
  1490.  
  1491.     if (e.user)
  1492.         e.user.display(text, e.code);
  1493.     else
  1494.         this.display(text, e.code);
  1495. }
  1496.  
  1497. CIRCNetwork.prototype.on330 = /* ircu's 330 numeric ("X is logged in as Y") */
  1498. function my_330 (e)
  1499. {
  1500.     var msg = getMsg(MSG_FMT_LOGGED_ON, [e.decodeParam(2), e.params[3]]);
  1501.     if (e.user)
  1502.         e.user.display(msg, "330");
  1503.     else
  1504.         this.display(msg, "330");
  1505. }
  1506.  
  1507. CIRCNetwork.prototype.on341 = /* invite reply */
  1508. function my_341 (e)
  1509. {
  1510.     this.display (getMsg(MSG_YOU_INVITE, [e.decodeParam(2), e.decodeParam(3)]),
  1511.                   "341");
  1512. }
  1513.  
  1514. CIRCNetwork.prototype.onInvite = /* invite message */
  1515. function my_invite (e)
  1516. {
  1517.     this.display(getMsg(MSG_INVITE_YOU, [e.user.unicodeName, e.user.name,
  1518.                                          e.user.host, e.channel.unicodeName]),
  1519.                  "INVITE");
  1520.  
  1521.     if ("messages" in e.channel)
  1522.         e.channel.join();
  1523. }
  1524.  
  1525. CIRCNetwork.prototype.on433 = /* nickname in use */
  1526. function my_433 (e)
  1527. {
  1528.     var nick = toUnicode(e.params[2], this);
  1529.  
  1530.     if ("pendingReclaimCheck" in this)
  1531.     {
  1532.         delete this.pendingReclaimCheck;
  1533.         return;
  1534.     }
  1535.  
  1536.     if (this.state == NET_CONNECTING)
  1537.     {
  1538.         // Force a number, thanks.
  1539.         var nickIndex = 1 * arrayIndexOf(this.prefs["nicknameList"], nick);
  1540.         var newnick;
  1541.  
  1542.         dd("433: failed with " + nick + " (" + nickIndex + ")");
  1543.  
  1544.         var tryList = true;
  1545.  
  1546.         if ((("_firstNick" in this) && (this._firstNick == -1)) ||
  1547.             (this.prefs["nicknameList"].length == 0) ||
  1548.             ((nickIndex != -1) && (this.prefs["nicknameList"].length < 2)))
  1549.         {
  1550.             tryList = false;
  1551.         }
  1552.  
  1553.         if (tryList)
  1554.         {
  1555.             nickIndex = (nickIndex + 1) % this.prefs["nicknameList"].length;
  1556.  
  1557.             if (("_firstNick" in this) && (this._firstNick == nickIndex))
  1558.             {
  1559.                 // We're back where we started. Give up with this method.
  1560.                 this._firstNick = -1;
  1561.                 tryList = false;
  1562.             }
  1563.         }
  1564.  
  1565.         if (tryList)
  1566.         {
  1567.             newnick = this.prefs["nicknameList"][nickIndex];
  1568.             dd("     trying " + newnick + " (" + nickIndex + ")");
  1569.  
  1570.             // Save first index we've tried.
  1571.             if (!("_firstNick" in this))
  1572.                 this._firstNick = nickIndex;
  1573.         }
  1574.         else
  1575.         {
  1576.             newnick = this.INITIAL_NICK + "_";
  1577.             dd("     trying " + newnick);
  1578.         }
  1579.  
  1580.         this.INITIAL_NICK = newnick;
  1581.         this.display(getMsg(MSG_RETRY_NICK, [nick, newnick]), "433");
  1582.         this.primServ.sendData("NICK " + fromUnicode(newnick, this) + "\n");
  1583.     }
  1584.     else
  1585.     {
  1586.         this.display(getMsg(MSG_NICK_IN_USE, nick), "433");
  1587.     }
  1588. }
  1589.  
  1590. CIRCNetwork.prototype.onStartConnect =
  1591. function my_sconnect (e)
  1592. {
  1593.     this.busy = true;
  1594.     updateProgress();
  1595.     if ("_firstNick" in this)
  1596.         delete this._firstNick;
  1597.  
  1598.     this.display (getMsg(MSG_CONNECTION_ATTEMPT,
  1599.                          [this.getURL(), e.server.getURL(), e.connectAttempt,
  1600.                           this.MAX_CONNECT_ATTEMPTS]), "INFO");
  1601. }
  1602.  
  1603. CIRCNetwork.prototype.onError =
  1604. function my_neterror (e)
  1605. {
  1606.     var msg;
  1607.     var type = "ERROR";
  1608.  
  1609.     if (typeof e.errorCode != "undefined")
  1610.     {
  1611.         switch (e.errorCode)
  1612.         {
  1613.             case JSIRC_ERR_NO_SOCKET:
  1614.                 msg = MSG_ERR_NO_SOCKET;
  1615.                 break;
  1616.  
  1617.             case JSIRC_ERR_EXHAUSTED:
  1618.                 msg = MSG_ERR_EXHAUSTED;
  1619.                 break;
  1620.  
  1621.             case JSIRC_ERR_OFFLINE:
  1622.                 msg = MSG_ERR_OFFLINE;
  1623.                 break;
  1624.  
  1625.             case JSIRC_ERR_NO_SECURE:
  1626.                 msg = getMsg(MSG_ERR_NO_SECURE, this.unicodeName);
  1627.                 break;
  1628.  
  1629.             case JSIRC_ERR_CANCELLED:
  1630.                 msg = MSG_ERR_CANCELLED;
  1631.                 type = "INFO";
  1632.                 break;
  1633.         }
  1634.     }
  1635.     else
  1636.         msg = e.params[e.params.length - 1];
  1637.  
  1638.     dispatch("sync-header");
  1639.     updateTitle();
  1640.  
  1641.     if (this.state == NET_OFFLINE)
  1642.     {
  1643.         this.busy = false;
  1644.         updateProgress();
  1645.     }
  1646.  
  1647.     this.display(msg, type);
  1648. }
  1649.  
  1650.  
  1651. CIRCNetwork.prototype.onDisconnect =
  1652. function my_netdisconnect (e)
  1653. {
  1654.     var msg;
  1655.     var msgType = "ERROR";
  1656.  
  1657.     if (typeof e.disconnectStatus != "undefined")
  1658.     {
  1659.         switch (e.disconnectStatus)
  1660.         {
  1661.             case 0:
  1662.                 msg = getMsg(MSG_CONNECTION_CLOSED,
  1663.                              [this.getURL(), e.server.getURL()]);
  1664.                 break;
  1665.  
  1666.             case NS_ERROR_CONNECTION_REFUSED:
  1667.                 msg = getMsg(MSG_CONNECTION_REFUSED,
  1668.                              [this.getURL(), e.server.getURL()]);
  1669.                 break;
  1670.  
  1671.             case NS_ERROR_NET_TIMEOUT:
  1672.                 msg = getMsg(MSG_CONNECTION_TIMEOUT,
  1673.                              [this.getURL(), e.server.getURL()]);
  1674.                 break;
  1675.  
  1676.             case NS_ERROR_NET_RESET:
  1677.                 msg = getMsg(MSG_CONNECTION_RESET,
  1678.                              [this.getURL(), e.server.getURL()]);
  1679.                 break;
  1680.  
  1681.             case NS_ERROR_UNKNOWN_HOST:
  1682.                 msg = getMsg(MSG_UNKNOWN_HOST,
  1683.                              [e.server.hostname, this.getURL(),
  1684.                               e.server.getURL()]);
  1685.                 break;
  1686.  
  1687.             default:
  1688.                 msg = getMsg(MSG_CLOSE_STATUS,
  1689.                              [this.getURL(), e.server.getURL(),
  1690.                               e.disconnectStatus]);
  1691.                 break;
  1692.         }
  1693.     }
  1694.     else
  1695.     {
  1696.         msg = getMsg(MSG_CONNECTION_CLOSED,
  1697.                      [this.getURL(), e.server.getURL()]);
  1698.     }
  1699.  
  1700.     // e.quitting signals the disconnect was intended: don't use "ERROR".
  1701.     if (e.quitting)
  1702.     {
  1703.         msgType = "DISCONNECT";
  1704.         msg = getMsg(MSG_CONNECTION_QUIT, [this.getURL(), e.server.getURL()]);
  1705.     }
  1706.  
  1707.     /* If we were only /trying/ to connect, and failed, just put an error on
  1708.      * the network tab. If we were actually connected ok, put it on all tabs.
  1709.      */
  1710.     if (this.state != NET_ONLINE)
  1711.     {
  1712.         this.busy = false;
  1713.         updateProgress();
  1714.  
  1715.         this.displayHere(msg, msgType);
  1716.     }
  1717.     else
  1718.     {
  1719.         for (var v in client.viewsArray)
  1720.         {
  1721.             var obj = client.viewsArray[v].source;
  1722.             if (obj != client)
  1723.             {
  1724.                 var details = getObjectDetails(obj);
  1725.                 if ("server" in details && details.server == e.server)
  1726.                     obj.displayHere(msg, msgType);
  1727.             }
  1728.         }
  1729.     }
  1730.  
  1731.     for (var c in this.primServ.channels)
  1732.     {
  1733.         var channel = this.primServ.channels[c];
  1734.         client.rdf.clearTargets(channel.getGraphResource(),
  1735.                                 client.rdf.resChanUser);
  1736.     }
  1737.  
  1738.     dispatch("sync-header");
  1739.     updateTitle();
  1740.     updateProgress();
  1741.     updateSecurityIcon();
  1742.  
  1743.     if ("userClose" in client && client.userClose &&
  1744.         client.getConnectionCount() == 0)
  1745.         window.close();
  1746.  
  1747.     if (("reconnect" in this) && this.reconnect)
  1748.     {
  1749.         this.connect(this.requireSecurity);
  1750.         delete this.reconnect;
  1751.     }
  1752. }
  1753.  
  1754. CIRCNetwork.prototype.onCTCPReplyPing =
  1755. function my_replyping (e)
  1756. {
  1757.     var delay = formatDateOffset((new Date() - new Date(Number(e.CTCPData))) /
  1758.                                  1000);
  1759.     this.display(getMsg(MSG_PING_REPLY, [e.user.unicodeName, delay]), "INFO",
  1760.                  e.user, "ME!");
  1761. }
  1762.  
  1763. CIRCNetwork.prototype.on221 =
  1764. CIRCNetwork.prototype.onUserMode =
  1765. function my_umode (e)
  1766. {
  1767.     if ("user" in e && e.user)
  1768.     {
  1769.         e.user.updateHeader();
  1770.         this.display(getMsg(MSG_USER_MODE, [e.user.unicodeName, e.params[2]]),
  1771.                      MT_MODE);
  1772.     }
  1773.     else
  1774.     {
  1775.         this.display(getMsg(MSG_USER_MODE, [e.params[1], e.params[2]]),
  1776.                      MT_MODE);
  1777.     }
  1778. }
  1779.  
  1780. CIRCNetwork.prototype.onNick =
  1781. function my_cnick (e)
  1782. {
  1783.     if (!ASSERT(userIsMe(e.user), "network nick event for third party"))
  1784.         return;
  1785.  
  1786.     if (getTabForObject(this))
  1787.     {
  1788.         this.displayHere(getMsg(MSG_NEWNICK_YOU, e.user.unicodeName),
  1789.                          "NICK", "ME!", e.user, this);
  1790.     }
  1791.  
  1792.     this.updateHeader();
  1793.     updateStalkExpression(this);
  1794. }
  1795.  
  1796. CIRCNetwork.prototype.onPing =
  1797. function my_netping (e)
  1798. {
  1799.     this.updateHeader(this);
  1800. }
  1801.  
  1802. CIRCNetwork.prototype.onPong =
  1803. function my_netpong (e)
  1804. {
  1805.     this.updateHeader(this);
  1806. }
  1807.  
  1808. CIRCNetwork.prototype.reclaimName =
  1809. function my_reclaimname()
  1810. {
  1811.     var network = this;
  1812.  
  1813.     function callback() {
  1814.         network.reclaimName();
  1815.     };
  1816.  
  1817.     if ("pendingReclaimCheck" in this)
  1818.         delete this.pendingReclaimCheck;
  1819.  
  1820.     // Function to attempt to get back the nickname the user wants.
  1821.     if ((this.state != NET_ONLINE) || !this.primServ)
  1822.         return false;
  1823.  
  1824.     if (this.primServ.me.unicodeName == this.prefs["nickname"])
  1825.         return false;
  1826.  
  1827.     this.reclaimLeft -= this.RECLAIM_WAIT;
  1828.  
  1829.     if (this.reclaimLeft <= 0)
  1830.         return false;
  1831.  
  1832.     this.pendingReclaimCheck = true;
  1833.     this.primServ.sendData("NICK " + fromUnicode(this.prefs["nickname"], this) + "\n");
  1834.  
  1835.     setTimeout(callback, this.RECLAIM_WAIT);
  1836.  
  1837.     return true;
  1838. }
  1839.  
  1840.  
  1841. /* We want to override the base implementations. */
  1842. CIRCChannel.prototype._join = CIRCChannel.prototype.join;
  1843. CIRCChannel.prototype._part = CIRCChannel.prototype.part;
  1844.  
  1845. CIRCChannel.prototype.join =
  1846. function chan_join(key)
  1847. {
  1848.     var joinFailedFn = function _joinFailedFn(t)
  1849.     {
  1850.         delete t.joinTimer;
  1851.         t.busy = false;
  1852.         updateProgress();
  1853.     }
  1854.     if (!this.joined)
  1855.     {
  1856.         this.joinTimer = setTimeout(joinFailedFn, 30000, this);
  1857.         this.busy = true;
  1858.         updateProgress();
  1859.     }
  1860.     this._join(key);
  1861. }
  1862.  
  1863. CIRCChannel.prototype.part =
  1864. function chan_part(reason)
  1865. {
  1866.     var partFailedFn = function _partFailedFn(t)
  1867.     {
  1868.         delete t.partTimer;
  1869.         t.busy = false;
  1870.         updateProgress();
  1871.     }
  1872.     this.partTimer = setTimeout(partFailedFn, 30000, this);
  1873.     this.busy = true;
  1874.     updateProgress();
  1875.     this._part(reason);
  1876. }
  1877.  
  1878. CIRCChannel.prototype.onInit =
  1879. function chan_oninit ()
  1880. {
  1881.     this.logFile = null;
  1882.     this.pendingNamesReply = false;
  1883. }
  1884.  
  1885. CIRCChannel.prototype.onPrivmsg =
  1886. function my_cprivmsg (e)
  1887. {
  1888.     var msg = e.decodeParam(2);
  1889.  
  1890.     this.display (msg, "PRIVMSG", e.user, this);
  1891. }
  1892.  
  1893. /* end of names */
  1894. CIRCChannel.prototype.on366 =
  1895. function my_366 (e)
  1896. {
  1897.     if (client.currentObject == this)
  1898.         /* hide the tree while we add (possibly tons) of nodes */
  1899.         client.rdf.setTreeRoot("user-list", client.rdf.resNullChan);
  1900.  
  1901.     client.rdf.clearTargets(this.getGraphResource(), client.rdf.resChanUser);
  1902.  
  1903.     var updates = new Array();
  1904.     for (var u in this.users)
  1905.     {
  1906.         this.users[u].updateGraphResource();
  1907.         this._addUserToGraph (this.users[u]);
  1908.         updates.push(this.users[u]);
  1909.     }
  1910.     this.addUsers(updates);
  1911.  
  1912.     if (client.currentObject == this)
  1913.         /* redisplay the tree */
  1914.         client.rdf.setTreeRoot("user-list", this.getGraphResource());
  1915.  
  1916.     if (this.pendingNamesReply)
  1917.     {
  1918.         this.parent.parent.display (e.channel.unicodeName + ": " +
  1919.                                     e.params[3], "366");
  1920.     }
  1921.     this.pendingNamesReply = false;
  1922. }
  1923.  
  1924. CIRCChannel.prototype.onTopic = /* user changed topic */
  1925. CIRCChannel.prototype.on332 = /* TOPIC reply */
  1926. function my_topic (e)
  1927. {
  1928.  
  1929.     if (e.code == "TOPIC")
  1930.         this.display (getMsg(MSG_TOPIC_CHANGED, [this.topicBy, this.topic]),
  1931.                       "TOPIC");
  1932.  
  1933.     if (e.code == "332")
  1934.     {
  1935.         if (this.topic)
  1936.         {
  1937.             this.display (getMsg(MSG_TOPIC,
  1938.                                  [this.unicodeName, this.topic]),
  1939.                           "TOPIC");
  1940.         }
  1941.         else
  1942.         {
  1943.             this.display (getMsg(MSG_NO_TOPIC, this.unicodeName), "TOPIC");
  1944.         }
  1945.     }
  1946.  
  1947.     this.updateHeader();
  1948.     updateTitle(this);
  1949.  
  1950. }
  1951.  
  1952. CIRCChannel.prototype.on333 = /* Topic setter information */
  1953. function my_topicinfo (e)
  1954. {
  1955.     this.display (getMsg(MSG_TOPIC_DATE, [this.unicodeName, this.topicBy,
  1956.                                           this.topicDate]), "TOPIC");
  1957. }
  1958.  
  1959. CIRCChannel.prototype.on353 = /* names reply */
  1960. function my_topic (e)
  1961. {
  1962.     if (this.pendingNamesReply)
  1963.     {
  1964.         this.parent.parent.display (e.channel.unicodeName + ": " +
  1965.                                     e.params[4], "NAMES");
  1966.     }
  1967. }
  1968.  
  1969. CIRCChannel.prototype.on367 = /* channel ban stuff */
  1970. CIRCChannel.prototype.on368 =
  1971. function my_bans(e)
  1972. {
  1973.     // Uh, we'll format this some other time.
  1974.     if (!("pendingBanList" in this))
  1975.         this.display(toUnicode(e.params.join(" "), this), e.code.toUpperCase());
  1976. }
  1977.  
  1978. CIRCChannel.prototype.on348 = /* channel except stuff */
  1979. CIRCChannel.prototype.on349 =
  1980. function my_excepts(e)
  1981. {
  1982.     if (!("pendingExceptList" in this))
  1983.         this.display(toUnicode(e.params.join(" "), this), e.code.toUpperCase());
  1984. }
  1985.  
  1986.  
  1987. CIRCChannel.prototype.onNotice =
  1988. function my_notice (e)
  1989. {
  1990.     this.display(e.decodeParam(2), "NOTICE", e.user, this);
  1991. }
  1992.  
  1993. CIRCChannel.prototype.onCTCPAction =
  1994. function my_caction (e)
  1995. {
  1996.     this.display (e.CTCPData, "ACTION", e.user, this);
  1997. }
  1998.  
  1999. CIRCChannel.prototype.onUnknownCTCP =
  2000. function my_unkctcp (e)
  2001. {
  2002.     this.display (getMsg(MSG_UNKNOWN_CTCP, [e.CTCPCode, e.CTCPData,
  2003.                                             e.user.unicodeName]),
  2004.                   "BAD-CTCP", e.user, this);
  2005. }
  2006.  
  2007. CIRCChannel.prototype.onJoin =
  2008. function my_cjoin (e)
  2009. {
  2010.     if (!("messages" in this))
  2011.         this.displayHere(getMsg(MSG_CHANNEL_OPENED, this.unicodeName), MT_INFO);
  2012.  
  2013.     if (userIsMe(e.user))
  2014.     {
  2015.         var params = [e.user.unicodeName, e.channel.unicodeName];
  2016.         this.display (getMsg(MSG_YOU_JOINED, params), "JOIN", 
  2017.                       e.server.me, this);
  2018.         if (client.globalHistory)
  2019.             client.globalHistory.addPage(this.getURL());
  2020.  
  2021.         if ("joinTimer" in this)
  2022.         {
  2023.             clearTimeout(this.joinTimer);
  2024.             delete this.joinTimer;
  2025.             this.busy = false;
  2026.             updateProgress();
  2027.         }
  2028.  
  2029.         /* !-channels are "safe" channels, and get a server-generated prefix.
  2030.          * For this reason, creating the channel is delayed until this point.
  2031.          */
  2032.         if (e.channel.unicodeName[0] == "!")
  2033.             dispatch("set-current-view", { view: e.channel });
  2034.     }
  2035.     else
  2036.     {
  2037.         this.display(getMsg(MSG_SOMEONE_JOINED,
  2038.                             [e.user.unicodeName, e.user.name, e.user.host,
  2039.                              e.channel.unicodeName]),
  2040.                      "JOIN", e.user, this);
  2041.     }
  2042.  
  2043.     this._addUserToGraph(e.user);
  2044.     /* We don't want to add ourself here, since the names reply we'll be
  2045.      * getting right after the join will include us as well! (FIXME)
  2046.      */
  2047.     if (!userIsMe(e.user))
  2048.         this.addUsers([e.user]);
  2049.     if (client.currentObject == this)
  2050.         updateUserList();
  2051.     this.updateHeader();
  2052. }
  2053.  
  2054. CIRCChannel.prototype.onPart =
  2055. function my_cpart (e)
  2056. {
  2057.     this._removeUserFromGraph(e.user);
  2058.     this.removeUsers([e.user]);
  2059.     this.updateHeader();
  2060.  
  2061.     if (userIsMe (e.user))
  2062.     {
  2063.         var params = [e.user.unicodeName, e.channel.unicodeName];
  2064.         this.display (getMsg(MSG_YOU_LEFT, params), "PART", e.user, this);
  2065.  
  2066.         if (client.currentObject == this)
  2067.             /* hide the tree while we remove (possibly tons) of nodes */
  2068.             client.rdf.setTreeRoot("user-list", client.rdf.resNullChan);
  2069.  
  2070.         client.rdf.clearTargets(this.getGraphResource(),
  2071.                                 client.rdf.resChanUser, true);
  2072.  
  2073.         if ("partTimer" in this)
  2074.         {
  2075.             clearTimeout(this.partTimer);
  2076.             delete this.partTimer;
  2077.             this.busy = false;
  2078.             updateProgress();
  2079.         }
  2080.  
  2081.         if (client.currentObject == this)
  2082.             /* redisplay the tree */
  2083.             client.rdf.setTreeRoot("user-list", this.getGraphResource());
  2084.  
  2085.         if ("noDelete" in this)
  2086.             delete this.noDelete;
  2087.         else if (client.prefs["deleteOnPart"])
  2088.             this.dispatch("delete");
  2089.     }
  2090.     else
  2091.     {
  2092.         if (e.reason)
  2093.         {
  2094.             this.display (getMsg(MSG_SOMEONE_LEFT_REASON,
  2095.                                  [e.user.unicodeName, e.channel.unicodeName,
  2096.                                   e.reason]),
  2097.                           "PART", e.user, this);
  2098.         }
  2099.         else
  2100.         {
  2101.             this.display (getMsg(MSG_SOMEONE_LEFT,
  2102.                                  [e.user.unicodeName, e.channel.unicodeName]),
  2103.                           "PART", e.user, this);
  2104.         }
  2105.     }
  2106. }
  2107.  
  2108. CIRCChannel.prototype.onKick =
  2109. function my_ckick (e)
  2110. {
  2111.     if (userIsMe (e.lamer))
  2112.     {
  2113.         if (e.user)
  2114.         {
  2115.             this.display (getMsg(MSG_YOURE_GONE,
  2116.                                  [e.lamer.unicodeName, e.channel.unicodeName,
  2117.                                   e.user.unicodeName, e.reason]),
  2118.                           "KICK", e.user, this);
  2119.         }
  2120.         else
  2121.         {
  2122.             this.display (getMsg(MSG_YOURE_GONE,
  2123.                                  [e.lamer.unicodeName, e.channel.unicodeName, 
  2124.                                   MSG_SERVER, e.reason]),
  2125.                           "KICK", (void 0), this);
  2126.         }
  2127.  
  2128.         /* Try 1 re-join attempt if allowed. */
  2129.         if (this.prefs["autoRejoin"])
  2130.             this.join(this.mode.key);
  2131.     }
  2132.     else
  2133.     {
  2134.         var enforcerProper, enforcerNick;
  2135.         if (e.user && userIsMe(e.user))
  2136.         {
  2137.             enforcerProper = "YOU";
  2138.             enforcerNick = "ME!";
  2139.         }
  2140.         else if (e.user)
  2141.         {
  2142.             enforcerProper = e.user.unicodeName;
  2143.             enforcerNick = e.user.encodedName;
  2144.         }
  2145.         else
  2146.         {
  2147.             enforcerProper = MSG_SERVER;
  2148.             enforcerNick = MSG_SERVER;
  2149.         }
  2150.  
  2151.         this.display (getMsg(MSG_SOMEONE_GONE,
  2152.                              [e.lamer.unicodeName, e.channel.unicodeName,
  2153.                               enforcerProper, e.reason]), "KICK", e.user, this);
  2154.     }
  2155.  
  2156.     this._removeUserFromGraph(e.lamer);
  2157.     this.removeUsers([e.lamer]);
  2158.     this.updateHeader();
  2159. }
  2160.  
  2161. CIRCChannel.prototype.onChanMode =
  2162. function my_cmode (e)
  2163. {
  2164.     if ("user" in e)
  2165.     {
  2166.         var msg = e.decodeParam(1);
  2167.         for (var i = 2; i < e.params.length; i++)
  2168.             msg += " " + e.decodeParam(i);
  2169.  
  2170.         this.display (getMsg(MSG_MODE_CHANGED, [msg, e.user.unicodeName]),
  2171.                       "MODE", e.user, this);
  2172.     }
  2173.  
  2174.     var updates = new Array();
  2175.     for (var u in e.usersAffected)
  2176.     {
  2177.         e.usersAffected[u].updateGraphResource();
  2178.         updates.push(e.usersAffected[u]);
  2179.     }
  2180.     this.updateUsers(updates);
  2181.  
  2182.     this.updateHeader();
  2183.     updateTitle(this);
  2184.     if (client.currentObject == this)
  2185.         updateUserList();
  2186. }
  2187.  
  2188. CIRCChannel.prototype.onNick =
  2189. function my_cnick (e)
  2190. {
  2191.     if (userIsMe (e.user))
  2192.     {
  2193.         if (getTabForObject(this))
  2194.         {
  2195.             this.displayHere(getMsg(MSG_NEWNICK_YOU, e.user.unicodeName),
  2196.                              "NICK", "ME!", e.user, this);
  2197.         }
  2198.         this.parent.parent.updateHeader();
  2199.     }
  2200.     else
  2201.     {
  2202.         this.display(getMsg(MSG_NEWNICK_NOTYOU, [e.oldNick, e.user.unicodeName]),
  2203.                      "NICK", e.user, this);
  2204.     }
  2205.  
  2206.     e.user.updateGraphResource();
  2207.     //this.updateUsers([e.user]);
  2208.     /* updateUsers isn't clever enough (currently) to handle a nick change, so
  2209.      * we fake the user leaving (with the old nick) and coming back (with the
  2210.      * new nick).
  2211.      */
  2212.     this.removeUsers([e.server.addUser(e.oldNick)]);
  2213.     this.addUsers([e.user]);
  2214.     if (client.currentObject == this)
  2215.         updateUserList();
  2216. }
  2217.  
  2218. CIRCChannel.prototype.onQuit =
  2219. function my_cquit (e)
  2220. {
  2221.     if (userIsMe(e.user))
  2222.     {
  2223.         /* I dont think this can happen */
  2224.         var pms = [e.user.unicodeName, e.server.parent.unicodeName, e.reason];
  2225.         this.display (getMsg(MSG_YOU_QUIT, pms),"QUIT", e.user, this);
  2226.     }
  2227.     else
  2228.     {
  2229.         this.display (getMsg(MSG_SOMEONE_QUIT,
  2230.                              [e.user.unicodeName, e.server.parent.unicodeName,
  2231.                               e.reason]),
  2232.                       "QUIT", e.user, this);
  2233.     }
  2234.  
  2235.     this._removeUserFromGraph(e.user);
  2236.     this.removeUsers([e.user]);
  2237.     this.updateHeader();
  2238. }
  2239.  
  2240. CIRCUser.prototype.onInit =
  2241. function user_oninit ()
  2242. {
  2243.     this.logFile = null;
  2244. }
  2245.  
  2246. CIRCUser.prototype.onPrivmsg =
  2247. function my_cprivmsg(e)
  2248. {
  2249.     if (!("messages" in this))
  2250.     {
  2251.         var limit = client.prefs["newTabLimit"];
  2252.         if (limit == 0 || client.viewsArray.length < limit)
  2253.             openQueryTab(e.server, e.user.unicodeName);
  2254.     }
  2255.  
  2256.     this.display(e.decodeParam(2), "PRIVMSG", e.user, e.server.me);
  2257. }
  2258.  
  2259. CIRCUser.prototype.onNick =
  2260. function my_unick (e)
  2261. {
  2262.     if (userIsMe(e.user))
  2263.     {
  2264.         this.parent.parent.updateHeader();
  2265.         updateTitle();
  2266.     }
  2267.     else if ("messages" in this && this.messages)
  2268.     {
  2269.         this.display(getMsg(MSG_NEWNICK_NOTYOU, [e.oldNick, e.user.unicodeName]),
  2270.                      "NICK", e.user, this);
  2271.     }
  2272.  
  2273.     this.updateHeader();
  2274.     var tab = getTabForObject(this);
  2275.     if (tab)
  2276.         tab.setAttribute("label", this.unicodeName);
  2277. }
  2278.  
  2279. CIRCUser.prototype.onNotice =
  2280. function my_notice (e)
  2281. {
  2282.     this.display(e.decodeParam(2), "NOTICE", this, e.server.me);
  2283. }
  2284.  
  2285. CIRCUser.prototype.onCTCPAction =
  2286. function my_uaction(e)
  2287. {
  2288.     if (!("messages" in this))
  2289.     {
  2290.         var limit = client.prefs["newTabLimit"];
  2291.         if (limit == 0 || client.viewsArray.length < limit)
  2292.             openQueryTab(e.server, e.user.unicodeName);
  2293.     }
  2294.  
  2295.     this.display(e.CTCPData, "ACTION", this, e.server.me);
  2296. }
  2297.  
  2298. CIRCUser.prototype.onUnknownCTCP =
  2299. function my_unkctcp (e)
  2300. {
  2301.     this.parent.parent.display (getMsg(MSG_UNKNOWN_CTCP,
  2302.                                        [e.CTCPCode, e.CTCPData,
  2303.                                         e.user.unicodeName]),
  2304.                                 "BAD-CTCP", this, e.server.me);
  2305. }
  2306.  
  2307. CIRCUser.prototype.onDCCChat =
  2308. function my_dccchat(e)
  2309. {
  2310.     if (!jsenv.HAS_SERVER_SOCKETS || !client.prefs["dcc.enabled"])
  2311.         return;
  2312.  
  2313.     var u = client.dcc.addUser(e.user, e.host);
  2314.     var c = client.dcc.addChat(u, e.port);
  2315.  
  2316.     client.munger.entries[".inline-buttons"].enabled = true;
  2317.     var cmds = getMsg(MSG_DCC_COMMAND_ACCEPT, "dcc-accept " + c.id) + " " +
  2318.                getMsg(MSG_DCC_COMMAND_DECLINE, "dcc-decline " + c.id);
  2319.     this.parent.parent.display(getMsg(MSG_DCCCHAT_GOT_REQUEST,
  2320.                                       [e.user.unicodeName, e.host, e.port, cmds]),
  2321.                                "DCC-CHAT");
  2322.     client.munger.entries[".inline-buttons"].enabled = false;
  2323.  
  2324.     // Pass the event over to the DCC Chat object.
  2325.     e.set = "dcc-chat";
  2326.     e.destObject = c;
  2327.     e.destMethod = "onGotRequest";
  2328. }
  2329.  
  2330. CIRCUser.prototype.onDCCSend =
  2331. function my_dccchat(e)
  2332. {
  2333.     if (!jsenv.HAS_SERVER_SOCKETS || !client.prefs["dcc.enabled"])
  2334.         return;
  2335.  
  2336.     var u = client.dcc.addUser(e.user, e.host);
  2337.     var f = client.dcc.addFileTransfer(u, e.port, e.file, e.size);
  2338.  
  2339.     client.munger.entries[".inline-buttons"].enabled = true;
  2340.     var cmds = getMsg(MSG_DCC_COMMAND_ACCEPT, "dcc-accept " + f.id) + " " +
  2341.                getMsg(MSG_DCC_COMMAND_DECLINE, "dcc-decline " + f.id);
  2342.     this.parent.parent.display(getMsg(MSG_DCCFILE_GOT_REQUEST,
  2343.                                       [e.user.unicodeName, e.host, e.port,
  2344.                                        e.file, e.size, cmds]),
  2345.                                "DCC-FILE");
  2346.     client.munger.entries[".inline-buttons"].enabled = false;
  2347.  
  2348.     // Pass the event over to the DCC Chat object.
  2349.     e.set = "dcc-file";
  2350.     e.destObject = f;
  2351.     e.destMethod = "onGotRequest";
  2352. }
  2353.  
  2354. CIRCUser.prototype.onDCCReject =
  2355. function my_dccchat(e)
  2356. {
  2357.     if (!client.prefs["dcc.enabled"])
  2358.         return;
  2359.  
  2360.     //FIXME: Uh... cope. //
  2361.  
  2362.     // Pass the event over to the DCC Chat object.
  2363.     //e.set = "dcc-file";
  2364.     //e.destObject = f;
  2365.     //e.destMethod = "onGotReject";
  2366. }
  2367.  
  2368. CIRCDCCChat.prototype.onInit =
  2369. function my_dccinit(e)
  2370. {
  2371.     /* FIXME: we're currently 'borrowing' the client views' prefs until we have
  2372.      * our own pref manager.
  2373.      */
  2374.     this.prefs = client.prefs;
  2375. }
  2376.  
  2377. CIRCDCCChat.prototype._getParams =
  2378. function my_dccgetparams()
  2379. {
  2380.     return [this.user.unicodeName, this.localIP, this.port];
  2381. }
  2382.  
  2383. CIRCDCCChat.prototype.onPrivmsg =
  2384. function my_dccprivmsg(e)
  2385. {
  2386.     this.displayHere(e.line, "PRIVMSG", e.user, "ME!");
  2387. }
  2388.  
  2389. CIRCDCCChat.prototype.onCTCPAction =
  2390. function my_uaction(e)
  2391. {
  2392.     this.displayHere(e.CTCPData, "ACTION", e.user, "ME!");
  2393. }
  2394.  
  2395. CIRCDCCChat.prototype.onUnknownCTCP =
  2396. function my_unkctcp(e)
  2397. {
  2398.     this.displayHere(getMsg(MSG_UNKNOWN_CTCP, [e.CTCPCode, e.CTCPData,
  2399.                                                e.user.unicodeName]),
  2400.                      "BAD-CTCP", e.user, "ME!");
  2401. }
  2402.  
  2403. CIRCDCCChat.prototype.onConnect =
  2404. function my_dccdisconnect(e)
  2405. {
  2406.     playEventSounds("dccchat", "connect");
  2407.     this.displayHere(getMsg(MSG_DCCCHAT_OPENED, this._getParams()), "DCC-CHAT");
  2408. }
  2409.  
  2410. CIRCDCCChat.prototype.onAbort =
  2411. function my_dccabort(e)
  2412. {
  2413.     this.display(getMsg(MSG_DCCCHAT_ABORTED, this._getParams()), "DCC-CHAT");
  2414. }
  2415.  
  2416. CIRCDCCChat.prototype.onFail =
  2417. function my_dccfail(e)
  2418. {
  2419.     this.display(getMsg(MSG_DCCCHAT_FAILED, this._getParams()), "DCC-CHAT");
  2420. }
  2421.  
  2422. CIRCDCCChat.prototype.onDisconnect =
  2423. function my_dccdisconnect(e)
  2424. {
  2425.     playEventSounds("dccchat", "disconnect");
  2426.     this.display(getMsg(MSG_DCCCHAT_CLOSED, this._getParams()), "DCC-CHAT");
  2427. }
  2428.  
  2429.  
  2430. CIRCDCCFileTransfer.prototype.onInit =
  2431. function my_dccfileinit(e)
  2432. {
  2433.     /* FIXME: we're currently 'borrowing' the client views' prefs until we have
  2434.      * our own pref manager.
  2435.      */
  2436.     this.prefs = client.prefs;
  2437.     this.busy = false;
  2438.     this.progress = -1;
  2439.     updateProgress();
  2440. }
  2441.  
  2442. CIRCDCCFileTransfer.prototype._getParams =
  2443. function my_dccfilegetparams()
  2444. {
  2445.     var dir = MSG_UNKNOWN;
  2446.  
  2447.     if (this.state.dir == DCC_DIR_GETTING)
  2448.         dir = MSG_DCCLIST_FROM;
  2449.  
  2450.     if (this.state.dir == DCC_DIR_SENDING)
  2451.         dir = MSG_DCCLIST_TO;
  2452.  
  2453.     return [this.filename, dir, this.user.unicodeName,
  2454.             this.localIP, this.port];
  2455. }
  2456.  
  2457. CIRCDCCFileTransfer.prototype.onConnect =
  2458. function my_dccfileconnect(e)
  2459. {
  2460.     this.displayHere(getMsg(MSG_DCCFILE_OPENED, this._getParams()), "DCC-FILE");
  2461.     this.busy = true;
  2462.     this.progress = 0;
  2463.     updateProgress();
  2464.     this._lastUpdate = new Date();
  2465.     this._lastPosition = 0;
  2466. }
  2467.  
  2468. CIRCDCCFileTransfer.prototype.onProgress =
  2469. function my_dccfileprogress(e)
  2470. {
  2471.     var now = new Date();
  2472.     var pcent = Math.floor(100 * this.position / this.size);
  2473.  
  2474.     // If we've moved 100KiB or waited 10s, update the progress bar.
  2475.     if ((this.position > this._lastPosition + 102400) ||
  2476.         (now - this._lastUpdate > 10000))
  2477.     {
  2478.         this.progress = pcent;
  2479.         updateProgress();
  2480.  
  2481.         var tab = getTabForObject(this);
  2482.         if (tab)
  2483.             tab.setAttribute("label", this.viewName + " (" + pcent + "%)");
  2484.  
  2485.         this.updateHeader();
  2486.  
  2487.         this._lastPosition = this.position;
  2488.     }
  2489.  
  2490.     // If it's also been 10s or more since we last displayed a msg...
  2491.     if (now - this._lastUpdate > 10000)
  2492.     {
  2493.         this.displayHere(getMsg(MSG_DCCFILE_PROGRESS,
  2494.                                 [pcent, this.position, this.size]),
  2495.                          "DCC-FILE");
  2496.  
  2497.         this._lastUpdate = now;
  2498.     }
  2499. }
  2500.  
  2501. CIRCDCCFileTransfer.prototype.onAbort =
  2502. function my_dccfileabort(e)
  2503. {
  2504.     this.busy = false;
  2505.     updateProgress();
  2506.     this.display(getMsg(MSG_DCCFILE_ABORTED, this._getParams()), "DCC-FILE");
  2507. }
  2508.  
  2509. CIRCDCCFileTransfer.prototype.onFail =
  2510. function my_dccfilefail(e)
  2511. {
  2512.     this.busy = false;
  2513.     updateProgress();
  2514.     this.display(getMsg(MSG_DCCFILE_FAILED, this._getParams()), "DCC-FILE");
  2515. }
  2516.  
  2517. CIRCDCCFileTransfer.prototype.onDisconnect =
  2518. function my_dccfiledisconnect(e)
  2519. {
  2520.     this.busy = false;
  2521.     updateProgress();
  2522.     this.updateHeader();
  2523.  
  2524.     var tab = getTabForObject(this);
  2525.     if (tab)
  2526.         tab.setAttribute("label", this.viewName + " (DONE)");
  2527.  
  2528.     this.display(getMsg(MSG_DCCFILE_CLOSED, this._getParams()), "DCC-FILE");
  2529. }
  2530.  
  2531.